<?php

namespace app\models\tableModel;

use app\entity\PageListEntity;
use app\models\tableModel\base\Position;
use app\service\tool\GenerateService;
use app\service\tool\UtilsService;
use yii\db\Expression;
use yii\db\ExpressionInterface;
use yii\helpers\ArrayHelper;

/**
 * [推荐位主数据]数据库操作模型
 * 作者: Editor Name
 * 日期: 2021/12/31
 * 时间: 14:11:15
 */
class PositionModel extends Position
{

    /**
     * 类型 列表
     * @var array
     */
    private static $typeList = [
        [
            'key'   => 'user',
            'value' => 1,
            'text'  => '用户'
        ],
        [
            'key'   => 'news',
            'value' => 2,
            'text'  => '文章'
        ]
    ];

    /**
     * 状态 列表
     * @var array
     */
    private static $statusList = [
        [
            'key'   => 'disabled',
            'value' => -1,
            'text'  => '禁用'
        ],
        [
            'key'   => 'open',
            'value' => 1,
            'text'  => '开启'
        ]
    ];

    /**
     * 排序最大值
     * @var int
     */
    protected static $maxSort = 999999;
    /**
     * 排序最小值
     * @var int
     */
    protected static $minSort = -999999;

    /**
     * 规则验证
     * @return array
     */
    public function rules()
    {

        // [类型]列表
        $typeList = array_column(self::getTypeList(), 'value');
        // [状态]列表
        $statusList = array_column(self::getStatusList(), 'value');

        $parent = parent::rules();
        return ArrayHelper::merge($parent, [
            ['type', 'in', 'range' => $typeList, 'message' => '类型不合法'],
            ['status', 'in', 'range' => $statusList, 'message' => '状态不合法'],
            ['sort', 'integer', 'max' => self::getMaxSort(), 'min' => self::getMinSort()],
        ]);
    }

    /**
     * 重写label的 文字
     */
    public function attributeLabels()
    {

        $parent = parent::attributeLabels();
        return array_merge($parent, [
            'id'          => '编号',
            'name'        => '名称',
            'max_num'     => '展示最大条目数',
            'sort'        => '排序',
            'type'        => '类型',
            'status'      => '状态',
            'add_time'    => '添加时间',
            'update_time' => '更新时间',
        ]);
    }

    /**
     * 场景
     * @return array
     */
    public function scenarios()
    {

        $scenarios = parent::scenarios();
        return ArrayHelper::merge($scenarios, [
            // 自定义场景 (无用请删除)
            'scUpdate' => [
                'someAttributes'
            ]
        ]);
    }

    /**
     * 自定义过滤单条[where]
     *  ` 仅限于通过方法[loadWhere]传入的条件, 可以在其中自定义逻辑
     *  ` 保证最终返回值为一维数组[Yii2]通用条件格式[运算符, 字段名, 值]
     * @param $condition array [sql]查询条件, 严格遵循Yii多条件查询:[运算符, 字段名, 值]
     * @return array
     */
    public function whereCustomFilter($condition = [])
    {

        // // 例：字段为编号 —— 操作符使用[IN]
        // if ($field == 'id') {
        //     return ['IN', $condition[1], $value[2]];
        // }

        // 最终返回必须为数组：[运算符, 字段名, 值]，符合Yii2规范
        return $condition;
    }

    /**
     * 获取分页信息
     * @param integer $page 当前页
     * @param integer $limit 获取几条
     * @param null $field 获取字段
     * @param array $opt 其他选项
     *  - hadDataFromIds —— array|false 加载子集是否存在推送条目编号
     * @return PageListEntity
     * @throws \yii\console\Exception
     */
    public function getPaginate($page, $limit, $field = null, $opt = [])
    {

        // 当前页面计算
        $page = max($page - 1, 0);

        // 查找的 字段空的 就默认给列表
        if (!$field) $field = '*';

        // 基础 where加载完毕
        $this->getBaseSql()->select($field);

        // 数据的获取 分页等
        $list = $this->getBaseSql()->offset($page * $limit)
            ->limit($limit)
            ->asArray()->all();

        // 格式化数据后的列表
        $list = $this->formatData($list, $field, $opt);
        // 总数量
        $total = $this->getBaseSql()->count();

        return PageListEntity::loadModel(compact('list', 'total'));
    }

    /**
     * 格式化列表活详情数据
     * @param array $list 列表
     * @param array|string $field 字段列表
     * @param array $opt 设置
     * @return mixed
     */
    public function formatData($list, $opt = [])
    {

        // 为空直接返回
        if (empty($list)) return $list;
        // 需要返回第一组（可能不是二维数组）
        $needFirst = false;
        if (!is_array(array_values($list)[0])) {
            $needFirst = true;
            $list      = [$list];
        }

        ### 某些参数初始化
        // 检测[推荐位附属数据][from_id]为条件是否存在
        $loadIsFromIds  = !empty($opt['hadDataFromIds']) && $opt['hadDataFromIds'];
        $hadDataFromIds = [];
        if ($loadIsFromIds) {
            $hadDataFromIds = array_column($list, 'id');
            $hadDataFromIds = PositionDataModel::loadModel(true)->loadWhere([
                'from_id' => $opt['hadDataFromIds'],
                'pos_id'  => $hadDataFromIds
            ])->getPaginate(1, 1, ['pos_id', 'id']);
            $hadDataFromIds = array_column($hadDataFromIds->getList(), null, 'pos_id');
        }


        // 格式化数据
        foreach ($list as $k => &$v) {

            // 更新时间
            if (isset($v['add_time'])) {
                $v['add_time_text']   = date('Y-m-d H:i:s', $v['add_time']);
                $v['add_time_text_s'] = date('Y-m-d', $v['add_time']);
            }

            // 更新时间
            if (isset($v['update_time'])) {
                $v['update_time_text']   = date('Y-m-d H:i:s', $v['update_time']);
                $v['update_time_text_s'] = date('Y-m-d', $v['update_time']);
            }

            // 类型 文本
            if (isset($v['type'])) {
                $v['type_text'] = self::getTypeText($v['type']);
            }

            // 状态 文本
            if (isset($v['status'])) {
                $v['status_text'] = self::getStatusText($v['status']);
            }

            // [推荐位附属数据][from_id]为条件是否存在
            if ($loadIsFromIds && !empty($hadDataFromIds[$v['id']])) {
                $v['had_item'] = 1;
            }
            // [推荐位附属数据][from_id]为条件是否存在 - 不存在
            else if ($loadIsFromIds) {
                $v['had_item'] = 0;
            }
        }

        reset($list);
        return $needFirst ? current($list) : $list;
    }

    /**
     * 添加|保存
     * @param bool $runValidation 是否仅仅运行验证，不运行保存
     * @param string[]|null|string $attributeNames 仅仅保存某几个字段
     * @return bool
     */
    public function save($runValidation = false, $attributeNames = null)
    {

        // 添加的话要赋值一些初始数据
        if (empty($this->id)) {

            // 可以是走[mongoId]
            $this->id = GenerateService::newMongoId();
        }

        $nowTime = time();
        // 添加时间
        if (empty($this->add_time)) $this->add_time = $nowTime;
        // 更新时间
        $this->update_time = $nowTime;
        // 检测
        if ($this->hasErrors() || !$this->validate($attributeNames)) {

            // 记录下错误日志
            \Yii::error([
                "`````````````````````````````````````````````````````````",
                "``                      数据库错误                       ``",
                "`` 错误详情: [[推荐位主数据]数据库操作模型]验证数据失败             ``",
                "`` 错误信息和参数详情:                                     ``",
                "`````````````````````````````````````````````````````````",
                $this->getAttributes(),
                $this->getErrors()
            ], 'db');
            return false;
        }

        // 需要 && 执行保存
        if (!$runValidation && !parent::save(false, $attributeNames)) {

            // 记录下错误日志
            \Yii::error([
                "`````````````````````````````````````````````````````````",
                "``                      数据库错误                       ``",
                "`` 错误详情: [[推荐位主数据]数据库操作模型]保存数据失败             ``",
                "`` 错误信息和参数详情:                                     ``",
                "`````````````````````````````````````````````````````````",
                $this->getAttributes(),
                $this->getErrors()
            ], 'db');
            return false;
        }

        return true;
    }

    /**
     * 保存推荐位数据条目
     * @param int $posId 推荐位编号
     * @param array $fromIdList 来源编号
     * @param int $fromType 来源类型
     * @return bool
     */
    public function savePosData($posId, $fromIdList, $fromType)
    {

        $posDataModel = null;
        // 循环以来源编号列表
        foreach ($fromIdList as $k => $v) {

            ### 获取推荐位条目信息
            $posDataModel = PositionDataModel::loadModel([
                'pos_id'    => $posId,
                'from_id'   => $v,
                'from_type' => $fromType
            ]);
            // 空的初始化
            if (!$posDataModel) {
                $posDataModel = PositionDataModel::loadModel();
            }
            // 赋值数据
            $posDataModel->load([
                'from_id'   => $v,
                'from_type' => $fromType,
                'pos_id'    => $this->id,
            ], '');
            // 加载数据
            if (!$posDataModel->loadData()) {

                $error = UtilsService::getModelError($posDataModel->errors);
                $this->addError($error['column'], $error['msg']);
                return false;
            }

            // 是否保存成功
            if (!$posDataModel->save(true)) {

                $error = UtilsService::getModelError($posDataModel->errors);
                $this->addError($error['column'], $error['msg']);
                return false;
            }
        }

        // 清理下条目
        if ($posDataModel && !$posDataModel->clearItem($this->max_num)) {

            $error = UtilsService::getModelError($posDataModel->errors);
            $this->addError($error['column'], $error['msg']);
            return false;
        }

        return true;
    }


    /**
     * [静态方法]批量快速更新某些字段
     * @param $condition
     * @param array $fieldVal
     * @return bool
     */
    public static function updateField($condition, $fieldVal = [])
    {

        // 验证字段
        $model = self::loadModel();
        $model->load($fieldVal, '');
        if (!$model->validate(array_keys($fieldVal))) {
            $error = UtilsService::getModelError($model->getErrors());
            self::$error_[$error['column']] = [$error['msg']];
            return false;
        }
        // 重新取值规则化后的特定值列表
        $fieldVal = $model->getAttributes(array_keys($fieldVal));

        $db = \Yii::$app->db->createCommand();

        try {

            $db->update(self::tableName(), $fieldVal, $condition)->execute();

            // 否则成功
            return true;
        } catch (\Exception $error) {

            // 记录下错误日志
            \Yii::error([
                "`````````````````````````````````````````````````````````",
                "``                      数据库错误                       ``",
                "`` 错误详情: [[推荐位主数据]数据库操作模型]批量修改[指定字段]失败，   ``",
                "``         {$error->getMessage()}                       ``",
                "`` SQL语句: {$db->getRawSql()}                         ``",
                "`` 错误信息和参数详情:                                     ``",
                "`````````````````````````````````````````````````````````",
                $error->getTraceAsString()
            ], 'db');

            // 静态错误
            self::$error_['db_error'] = empty(self::$error_['db_error']) ?
                [$error->getMessage()] : array_merge(self::$error_['db_error'], [$error->getMessage()]);

            return false;
        }
    }

    /**
     * 批量添加数据
     * @param array $createData
     * @return bool
     */
    public static function createData($createData = [])
    {

        $db = \Yii::$app->db->createCommand();
        try {

            ### 先数据格式化
            // 如果是一维数组则二维数组化
            if (ArrayHelper::isAssociative($createData)) {
                $createData = [$createData];
            }
            $values = [];
            foreach ($createData as $k => $v) {

                $model = self::loadModel();
                $model->load($createData[$k], '');
                if (!$model->save(true)) {
                    // 取出错误信息
                    $error = UtilsService::getModelError($model->errors);
                    // 添加到静态方法上
                    self::$error_[$error['column']] = [$error['msg']];
                    return false;
                }

                $createData[$k] = $model->getAttributes($model::getTableSchema()->getColumnNames());

                // 循环一些数据
                foreach ($createData[$k] as $kc => $vc) {
                    // 字段类型为[JSON]类型需要转为[JSON]
                    if (is_array($vc)) {
                        $createData[$k][$kc] = json_encode($vc, JSON_UNESCAPED_UNICODE);
                        continue;
                    }
                }

                // 值赋值
                $values[] = array_values($createData[$k]);;
            }

            ### 取出此次操作的字段列表
            $columns = !current($createData) ? [] : array_keys(current($createData));

            // 执行
            $addResult = $db->batchInsert(self::tableName(), $columns, $values)->execute();

            return $addResult;
        } catch (\Exception $error) {

            // 记录下错误日志
            \Yii::error([

                "`````````````````````````````````````````````````````````",
                "``                      数据库错误                       ``",
                "`` 错误详情: [[推荐位主数据]数据库操作模型]批量添加[数据]失败，      ``",
                "``         {$error->getMessage()}                       ``",
                "`` SQL语句: {$db->getRawSql()}                         ``",
                "`` 错误信息和参数详情:                                     ``",
                "`````````````````````````````````````````````````````````",
                $error->getTraceAsString()
            ], 'db');

            // 静态错误
            self::$error_['db_error'] = empty(self::$error_['db_error']) ?
                [$error->getMessage()] : array_merge(self::$error_['db_error'], [$error->getMessage()]);

            return false;
        }
    }

    /**
     * 更新某些字段自增|自减
     * @param $condition
     * @param array $fieldVal 增/减加的字段
     * @return bool
     */
    public static function updateCounter($condition, $fieldVal = [])
    {

        $model = new self();
        foreach ($fieldVal as $k => $v) {

            if (!$model->hasAttribute($k)) {

                unset($fieldVal[$k]);
                continue;
            }
        }

        try {

            $model->updateAllCounters($fieldVal, $condition);

            // 否则成功
            return true;
        } catch (\Exception $error) {

            // 记录下错误日志
            \Yii::error([

                "`````````````````````````````````````````````````````````",
                "``                      数据库错误                       ``",
                "`` 错误详情: [[推荐位主数据]数据库操作模型]批量增/减[指定字段]失败   ``",
                "``         {$error->getMessage()}                       ``",
                "`` 错误信息和参数详情:                                     ``",
                "`````````````````````````````````````````````````````````",
                $error->getTraceAsString()
            ], 'db');

            // 静态错误
            self::$error_['db_error'] = empty(self::$error_['db_error']) ?
                [$error->getMessage()] : array_merge(self::$error_['db_error'], [$error->getMessage()]);

            return false;
        }
    }


    /**
     * 返回排序最大值
     * @return int
     */
    public static function getMaxSort()
    {
        return self::$maxSort;
    }

    /**
     * 返回排序最小值
     * @return int
     */
    public static function getMinSort()
    {
        return self::$minSort;
    }


    /**
     * 获取[类型]用户 值
     * @return mixed
     */
    public static function getTypeUser()
    {

        $list = array_column(self::$typeList, null, 'key');
        return $list['user']['value'];
    }

    /**
     * 获取[类型]文章 值
     * @return mixed
     */
    public static function getTypeNews()
    {

        $list = array_column(self::$typeList, null, 'key');
        return $list['news']['value'];
    }

    /**
     * 获取[类型]文本
     * @param $value
     * @return mixed|string
     */
    public static function getTypeText($value)
    {

        // 列表
        $list = array_column(self::$typeList, null, 'value');
        // 不合法 - 不存在
        if (empty($list[$value]['text'])) return '--';

        // 最终正常返回
        return $list[$value]['text'];
    }

    /**
     * 获取[类型]列表 值
     * @return mixed|string
     */
    public static function getTypeList()
    {

        // 最终正常返回
        return self::$typeList;
    }


    /**
     * 获取[状态][关闭]值
     * @return mixed
     */
    public static function getStatusDefault()
    {

        $list = array_column(self::$statusList, null, 'key');
        return $list['default']['value'];
    }

    /**
     * 获取[状态][关闭]值
     * @return mixed
     */
    public static function getStatusDisabled()
    {

        $list = array_column(self::$statusList, null, 'key');
        return $list['disabled']['value'];
    }

    /**
     * 获取[状态][开启]值
     * @return mixed
     */
    public static function getStatusOpen()
    {

        $list = array_column(self::$statusList, null, 'key');
        return $list['open']['value'];
    }

    /**
     * 获取[状态]文本
     * @param $value
     * @return mixed|string
     */
    public static function getStatusText($value)
    {

        // 列表
        $list = array_column(self::$statusList, null, 'value');
        // 不合法 - 不存在
        if (empty($list[$value]['text'])) return '--';

        // 最终正常返回
        return $list[$value]['text'];
    }

    /**
     * 获取[状态]列表 值
     * @return mixed|string
     */
    public static function getStatusList()
    {

        // 最终正常返回
        return self::$statusList;
    }


    /**
     * 获取静态错误
     * @return mixed
     */
    public static function getStaticErrors()
    {
        return self::$error_;
    }
}
